home *** CD-ROM | disk | FTP | other *** search
/ Super Shareware Collection / Super Shareware Collection.iso / os_2 / clisp.zip / PLACES.LSP < prev    next >
Lisp/Scheme  |  1994-02-05  |  50KB  |  1,040 lines

  1. ; CLISP - PLACES.LSP
  2. ; CLISP-spezifisch: string-concat, %rplaca, %rplacd, store, %setelt, ...
  3.  
  4. (in-package "SYSTEM")
  5. ;-------------------------------------------------------------------------------
  6. ; Funktionen zur Definition und zum Ausnutzen von places:
  7. ;-------------------------------------------------------------------------------
  8. (defun setf-symbol (symbol) ; liefert uninterniertes Symbol für SYSTEM::SETF-FUNCTION
  9.   (make-symbol
  10.     (string-concat
  11.       "(SETF "
  12.       (let ((pack (symbol-package symbol))) (if pack (package-name pack) "#"))
  13.       ":"
  14.       (symbol-name symbol)
  15.       ")"
  16. ) ) )
  17. ;-------------------------------------------------------------------------------
  18. (defun get-setf-symbol (symbol) ; liefert das Symbol bei SYSTEM::SETF-FUNCTION
  19.   (or (get symbol 'SYSTEM::SETF-FUNCTION)
  20.       (progn
  21.         (when (get symbol 'SYSTEM::SETF-EXPANDER)
  22.           (warn #+DEUTSCH "Die Funktion (~S ~S) ist durch einen SETF-Expander verborgen."
  23.                 #+ENGLISH "The function (~S ~S) is hidden by a SETF expander."
  24.                 #+FRANCAIS "La fonction (~S ~S) est cachée par une méthode SETF."
  25.                 'setf symbol
  26.         ) )
  27.         (setf (get symbol 'SYSTEM::SETF-FUNCTION) (setf-symbol symbol))
  28. ) )   )
  29. ;-------------------------------------------------------------------------------
  30. (defun get-funname-symbol (funname) ; Abbildung Funktionsname --> Symbol
  31.   (if (atom funname)
  32.     funname
  33.     (get-setf-symbol (second funname))
  34. ) )
  35. ;-------------------------------------------------------------------------------
  36. (defun get-setf-method-multiple-value (form &optional (env (vector nil nil)))
  37.   (loop
  38.     ; 1. Schritt: nach globalen SETF-Definitionen suchen:
  39.     (when (and (consp form) (symbolp (car form)))
  40.       (when (global-in-fenv-p (car form) (svref env 1))
  41.         ; Operator nicht lokal definiert
  42.         (let ((plist-info (get (first form) 'SYSTEM::SETF-EXPANDER)))
  43.           (when plist-info
  44.             (return-from get-setf-method-multiple-value
  45.               (if (symbolp plist-info) ; Symbol kommt von kurzem DEFSETF
  46.                 (do* ((storevar (gensym))
  47.                       (tempvars nil (cons (gensym) tempvars))
  48.                       (tempforms nil (cons (car formr) tempforms))
  49.                       (formr (cdr form) (cdr formr)))
  50.                      ((atom formr)
  51.                       (setq tempforms (nreverse tempforms))
  52.                       (values tempvars
  53.                               tempforms
  54.                               `(,storevar)
  55.                               `(,plist-info ,@tempvars ,storevar)
  56.                               `(,(first form) ,@tempvars)
  57.                      ))
  58.                 )
  59.                 (let ((argcount (car plist-info)))
  60.                   (if (eql argcount -5)
  61.                     ; (-5 . fun) kommt von DEFINE-SETF-METHOD
  62.                     (funcall (cdr plist-info) form env)
  63.                     ; (argcount . fun) kommt von langem DEFSETF
  64.                     (let ((access-form form)
  65.                           (tempvars '())
  66.                           (tempforms '())
  67.                           (new-access-form '()))
  68.                       (let ((i 0)) ; Argumente-Zähler
  69.                         ; argcount = -1 falls keine Keyword-Argumente existieren
  70.                         ; bzw.     = Anzahl der einzelnen Argumente vor &KEY,
  71.                         ;          = nil nachdem diese abgearbeitet sind.
  72.                         (dolist (argform (cdr access-form))
  73.                           (when (eql i argcount) (setf argcount nil i 0))
  74.                           (if (and (null argcount) (evenp i))
  75.                             (if (keywordp argform)
  76.                               (push argform new-access-form)
  77.                               (error #+DEUTSCH "Das Argument ~S zu ~S sollte ein Keyword sein."
  78.                                      #+ENGLISH "The argument ~S to ~S should be a keyword."
  79.                                      #+FRANCAIS "L'argument ~S de ~S doit être un mot-clé."
  80.                                      argform (car access-form)
  81.                             ) )
  82.                             (let ((tempvar (gensym)))
  83.                               (push tempvar tempvars)
  84.                               (push argform tempforms)
  85.                               (push tempvar new-access-form)
  86.                           ) )
  87.                           (incf i)
  88.                       ) )
  89.                       (setq new-access-form
  90.                         (cons (car access-form) (nreverse new-access-form))
  91.                       )
  92.                       (let ((newval-var (gensym)))
  93.                         (values
  94.                           (nreverse tempvars)
  95.                           (nreverse tempforms)
  96.                           (list newval-var)
  97.                           (funcall (cdr plist-info) new-access-form newval-var)
  98.                           new-access-form
  99.                 ) ) ) ) )
  100.             ) )
  101.     ) ) ) )
  102.     ; 2. Schritt: macroexpandieren
  103.     (when (eq form (setq form (macroexpand-1 form env)))
  104.       (return)
  105.   ) )
  106.   ; 3. Schritt: Default-SETF-Methoden
  107.   (cond ((symbolp form)
  108.          (return-from get-setf-method-multiple-value
  109.            (let ((storevar (gensym)))
  110.              (values nil
  111.                      nil
  112.                      `(,storevar)
  113.                      `(SETQ ,form ,storevar)
  114.                      `,form
  115.         )) ) )
  116.         ((and (consp form) (symbolp (car form)))
  117.          (return-from get-setf-method-multiple-value
  118.            (do* ((storevar (gensym))
  119.                  (tempvars nil (cons (gensym) tempvars))
  120.                  (tempforms nil (cons (car formr) tempforms))
  121.                  (formr (cdr form) (cdr formr)))
  122.                 ((atom formr)
  123.                  (setq tempforms (nreverse tempforms))
  124.                  (values tempvars
  125.                          tempforms
  126.                          `(,storevar)
  127.                          `((SETF ,(first form)) ,storevar ,@tempvars)
  128.                          `(,(first form) ,@tempvars)
  129.                 ))
  130.         )) )
  131.         (t (error #+DEUTSCH "Das Argument muß eine 'SETF-place' sein, ist aber keine: ~S"
  132.                   #+ENGLISH "Argument ~S is not a SETF place."
  133.                   #+FRANCAIS "L'argument ~S doit représenter une place modifiable."
  134.                   form
  135.   )     )  )
  136. )
  137. ;-------------------------------------------------------------------------------
  138. (defun get-setf-method (form &optional (env (vector nil nil)))
  139.   (multiple-value-bind (vars vals stores store-form access-form)
  140.       (get-setf-method-multiple-value form env)
  141.     (unless (and (consp stores) (null (cdr stores)))
  142.       (error #+DEUTSCH "Diese 'SETF-place' produziert mehrere 'Store-Variable': ~S"
  143.              #+ENGLISH "SETF place ~S produces more than one store variable."
  144.              #+FRANCAIS "La place modifiable ~S produit plusieurs variables de résultat."
  145.              form
  146.     ) )
  147.     (values vars vals stores store-form access-form)
  148. ) )
  149. ;-------------------------------------------------------------------------------
  150. ; In einfachen Zuweisungen wie (SETQ foo #:G0) darf #:G0 direkt ersetzt werden.
  151. (defun simple-assignment-p (store-form stores)
  152.   (and (eql (length stores) 1)
  153.        (consp store-form)
  154.        (eq (first store-form) 'SETQ)
  155.        (eql (length store-form) 3)
  156.        (symbolp (second store-form))
  157.        (simple-use-p (third store-form) (first stores))
  158. ) )
  159. (defun simple-use-p (form var)
  160.   (or (eq form var)
  161.       (and (consp form) (eq (first form) 'THE) (eql (length form) 3)
  162.            (simple-use-p (third form) var)
  163. ) )   )
  164. ;-------------------------------------------------------------------------------
  165. (defun documentation (symbol doctype)
  166.   (unless (function-name-p symbol)
  167.     (error #+DEUTSCH "~S: Das ist als erstes Argument unzulässig, da kein Symbol: ~S"
  168.            #+ENGLISH "~S: first argument ~S is illegal, not a symbol"
  169.            #+FRANCAIS "~S : Le premier argument ~S est invalide car ce n'est pas un symbole."
  170.            'documentation symbol
  171.   ) )
  172.   (getf (get (get-funname-symbol symbol) 'SYSTEM::DOCUMENTATION-STRINGS) doctype)
  173. )
  174. (defun SYSTEM::%SET-DOCUMENTATION (symbol doctype value)
  175.   (unless (function-name-p symbol)
  176.     (error #+DEUTSCH "~S: Das ist als erstes Argument unzulässig, da kein Symbol: ~S"
  177.            #+ENGLISH "~S: first argument ~S is illegal, not a symbol"
  178.            #+FRANCAIS "~S : Le premier argument ~S est invalide car ce n'est pas un symbole."
  179.            'documentation symbol
  180.   ) )
  181.   (setq symbol (get-funname-symbol symbol))
  182.   (if (null value)
  183.     (when (getf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype)
  184.       (remf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype)
  185.       nil
  186.     )
  187.     (setf (getf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype) value)
  188. ) )
  189. ;-------------------------------------------------------------------------------
  190. (defmacro push (item place &environment env)
  191.   (let ((itemvar (gensym)))
  192.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  193.       (let ((bindlist (mapcar #'list SM1 SM2)))
  194.         (if bindlist
  195.           (push `(,itemvar ,item) bindlist)
  196.           (setq itemvar item)
  197.         )
  198.         (let ((valform `(CONS ,itemvar ,SM5)))
  199.           (if (simple-assignment-p SM4 SM3)
  200.             (setq SM4 (subst valform (first SM3) SM4))
  201.             (setq bindlist (nconc bindlist `((,(first SM3) ,valform))))
  202.           )
  203.           (if bindlist
  204.             `(LET* ,bindlist
  205.                ,SM4
  206.              )
  207.             SM4
  208.           )
  209. ) ) ) ) )
  210. ;-------------------------------------------------------------------------------
  211. (defmacro define-setf-method (accessfn lambdalist &body body &environment env)
  212.   (unless (symbolp accessfn)
  213.     (error #+DEUTSCH "Der Name der Access-Function muß ein Symbol sein und nicht ~S."
  214.            #+ENGLISH "The name of the access function must be a symbol, not ~S"
  215.            #+FRANCAIS "Le nom de la fonction d'accès doit être un symbole et non ~S."
  216.            accessfn
  217.   ) )
  218.   (multiple-value-bind (body-rest declarations docstring)
  219.       (system::parse-body body t env)
  220.     (if (null body-rest) (setq body-rest '(NIL)))
  221.     (let ((name (make-symbol (string-concat "SETF-" (symbol-name accessfn)))))
  222.       (multiple-value-bind (newlambdalist envvar) (remove-env-arg lambdalist name)
  223.         (let ((SYSTEM::%ARG-COUNT 0)
  224.               (SYSTEM::%MIN-ARGS 0)
  225.               (SYSTEM::%RESTP nil)
  226.               (SYSTEM::%LET-LIST nil)
  227.               (SYSTEM::%KEYWORD-TESTS nil)
  228.               (SYSTEM::%DEFAULT-FORM nil)
  229.              )
  230.           (SYSTEM::ANALYZE1 newlambdalist '(CDR SYSTEM::%LAMBDA-LIST)
  231.                             name 'SYSTEM::%LAMBDA-LIST
  232.           )
  233.           (if (null newlambdalist)
  234.             (push `(IGNORE SYSTEM::%LAMBDA-LIST) declarations)
  235.           )
  236.           (let ((lengthtest (sys::make-length-test 'SYSTEM::%LAMBDA-LIST))
  237.                 (mainform
  238.                   `(LET* ,(nreverse SYSTEM::%LET-LIST)
  239.                      ,@(if declarations `(,(cons 'DECLARE declarations)))
  240.                      ,@SYSTEM::%KEYWORD-TESTS
  241.                      ,@body-rest
  242.                    )
  243.                ))
  244.             (if lengthtest
  245.               (setq mainform
  246.                 `(IF ,lengthtest
  247.                    (ERROR #+DEUTSCH "Der SETF-Expander für ~S kann nicht mit ~S Argumenten aufgerufen werden."
  248.                           #+ENGLISH "The SETF expander for ~S may not be called with ~S arguments."
  249.                           #+FRANCAIS "L'«expandeur» SETF pour ~S ne peut pas être appelé avec ~S arguments."
  250.                           (QUOTE ,accessfn) (1- (LENGTH SYSTEM::%LAMBDA-LIST))
  251.                    )
  252.                    ,mainform
  253.               )  )
  254.             )
  255.             `(EVAL-WHEN (LOAD COMPILE EVAL)
  256.                (LET ()
  257.                  (DEFUN ,name (SYSTEM::%LAMBDA-LIST ,(or envvar 'SYSTEM::ENV))
  258.                    ,@(if envvar '() '((DECLARE (IGNORE SYSTEM::ENV))))
  259.                    ,mainform
  260.                  )
  261.                  (SYSTEM::%PUT ',accessfn 'SYSTEM::SETF-EXPANDER
  262.                    (CONS -5 (FUNCTION ,name))
  263.                  )
  264.                  (SYSTEM::%SET-DOCUMENTATION ',accessfn 'SETF ',docstring)
  265.                  ',accessfn
  266.              ) )
  267. ) ) ) ) ) )
  268. ;-------------------------------------------------------------------------------
  269. (defmacro defsetf (accessfn &rest args &environment env)
  270.   (cond ((and (consp args) (not (listp (first args))) (symbolp (first args)))
  271.          `(EVAL-WHEN (LOAD COMPILE EVAL)
  272.             (LET ()
  273.               (SYSTEM::%PUT ',accessfn 'SYSTEM::SETF-EXPANDER ',(first args))
  274.               (SYSTEM::%SET-DOCUMENTATION ',accessfn 'SETF
  275.                 ,(if (and (null (cddr args))
  276.                           (or (null (second args)) (stringp (second args)))
  277.                      )
  278.                    (second args)
  279.                    (if (cddr args)
  280.                      (error #+DEUTSCH "Zu viele Argumente für DEFSETF: ~S"
  281.                             #+ENGLISH "Too many arguments to DEFSETF: ~S"
  282.                             #+FRANCAIS "Trop d'arguments pour DEFSETF : ~S"
  283.                             (cdr args)
  284.                      )
  285.                      (error #+DEUTSCH "Der Dok.-String zu DEFSETF muß ein String sein: ~S"
  286.                             #+ENGLISH "The doc string to DEFSETF must be a string: ~S"
  287.                             #+FRANCAIS "La documentation pour DEFSETF doit être un chaîne : ~S"
  288.                             (second args)
  289.                  ) ) )
  290.               )
  291.               ',accessfn
  292.           ) )
  293.         )
  294.         ((and (consp args) (listp (first args)) (consp (cdr args)) (listp (second args)))
  295.          (cond ((= (length (second args)) 1))
  296.                ((= (length (second args)) 0)
  297.                 (error #+DEUTSCH "Bei DEFSETF muß genau eine 'Store-Variable' angegeben werden."
  298.                        #+ENGLISH "Missing store variable in DEFSETF."
  299.                        #+FRANCAIS "Une variable de résultat doit être précisée dans DEFSETF."
  300.                ))
  301.                (t (cerror #+DEUTSCH "Die überzähligen Variablen werden ignoriert."
  302.                           #+ENGLISH "The excess variables will be ignored."
  303.                           #+FRANCAIS "Les variables en excès seront ignorées."
  304.                           #+DEUTSCH "Bei DEFSETF ist nur eine 'Store-Variable' erlaubt."
  305.                           #+ENGLISH "Only one store variable is allowed in DEFSETF."
  306.                           #+FRANCAIS "Une seule variable de résultat est permise dans DEFSETF."
  307.          )     )  )
  308.          (multiple-value-bind (body-rest declarations docstring)
  309.              (system::parse-body (cddr args) t env)
  310.            (let* (arg-count
  311.                   (setter
  312.                     (let* ((lambdalist (first args))
  313.                            (storevar (first (second args)))
  314.                            (SYSTEM::%ARG-COUNT 0)
  315.                            (SYSTEM::%MIN-ARGS 0)
  316.                            (SYSTEM::%RESTP nil)
  317.                            (SYSTEM::%LET-LIST nil)
  318.                            (SYSTEM::%KEYWORD-TESTS nil)
  319.                            (SYSTEM::%DEFAULT-FORM nil))
  320.                       (SYSTEM::ANALYZE1 lambdalist '(CDR SYSTEM::%ACCESS-ARGLIST)
  321.                                         accessfn 'SYSTEM::%ACCESS-ARGLIST
  322.                       )
  323.                       (setq arg-count (if (member '&KEY lambdalist) SYSTEM::%ARG-COUNT -1))
  324.                       `(LAMBDA (SYSTEM::%ACCESS-ARGLIST ,storevar)
  325.                          ,@(if (null lambdalist)
  326.                              `((DECLARE (IGNORE SYSTEM::%ACCESS-ARGLIST)))
  327.                            )
  328.                          (LET* ,(nreverse SYSTEM::%LET-LIST)
  329.                            ,@(if declarations `(,(cons 'DECLARE declarations)))
  330.                            ,@SYSTEM::%KEYWORD-TESTS
  331.                            ,@body-rest
  332.                        ) )
  333.                  )) )
  334.              `(EVAL-WHEN (LOAD COMPILE EVAL)
  335.                 (LET ()
  336.                   (SYSTEM::%PUT ',accessfn 'SYSTEM::SETF-EXPANDER
  337.                     (CONS ,arg-count
  338.                           (FUNCTION ,(concat-pnames "SETF-" accessfn) ,setter)
  339.                   ) )
  340.                   (SYSTEM::%SET-DOCUMENTATION ',accessfn 'SETF ,docstring)
  341.                   ',accessfn
  342.               ) )
  343.         )) )
  344.         (t (error #+DEUTSCH "DEFSETF-Aufruf für ~S ist falsch aufgebaut."
  345.                   #+ENGLISH "Illegal syntax in DEFSETF for ~S"
  346.                   #+FRANCAIS "Le DEFSETF ~S est mal formé."
  347.                   accessfn
  348. ) )     )  )
  349. ;-------------------------------------------------------------------------------
  350. (defmacro pop (place &environment env)
  351.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  352.     (do* ((SM1r SM1 (cdr SM1r))
  353.           (SM2r SM2 (cdr SM2r))
  354.           (bindlist nil))
  355.          ((null SM1r)
  356.           (let* ((valform
  357.                    (if (and (symbolp SM5) (simple-assignment-p SM4 SM3))
  358.                      SM5
  359.                      (progn (push `(,(first SM3) ,SM5) bindlist) (first SM3))
  360.                  ) )
  361.                  (newvalform `(CDR ,valform))
  362.                  (form `(PROG1
  363.                           (CAR ,valform)
  364.                           ,@(if (simple-assignment-p SM4 SM3)
  365.                               (list (subst newvalform (first SM3) SM4))
  366.                               (list `(SETQ ,(first SM3) ,newvalform) SM4)
  367.                             )
  368.                         )
  369.                 ))
  370.             (if bindlist
  371.               `(LET* ,(nreverse bindlist) ,form)
  372.               form
  373.          )) )
  374.       (push `(,(first SM1r) ,(first SM2r)) bindlist)
  375. ) ) )
  376. ;-------------------------------------------------------------------------------
  377. (defmacro psetf (&whole form &rest args &environment env)
  378.   (do ((arglist args (cddr arglist))
  379.        (bindlist nil)
  380.        (storelist nil))
  381.       ((atom arglist)
  382.        `(LET* ,(nreverse bindlist)
  383.           ,@storelist
  384.           NIL
  385.       ) )
  386.     (when (atom (cdr arglist))
  387.       (error #+DEUTSCH "~S mit einer ungeraden Zahl von Argumenten aufgerufen: ~S"
  388.              #+ENGLISH "~S called with an odd number of arguments: ~S"
  389.              #+FRANCAIS "~S fut appelé avec un nombre impair d'arguments : ~S"
  390.              'psetf form
  391.     ) )
  392.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method (first arglist) env)
  393.       (declare (ignore SM5))
  394.       (do* ((SM1r SM1 (cdr SM1r))
  395.             (SM2r SM2 (cdr SM2r)))
  396.            ((null SM1r))
  397.         (push `(,(first SM1r) ,(first SM2r)) bindlist)
  398.       )
  399.       (push `(,(first SM3) ,(second arglist)) bindlist)
  400.       (push SM4 storelist)
  401. ) ) )
  402. ;-------------------------------------------------------------------------------
  403. (defmacro pushnew (item place &rest keylist &environment env)
  404.   (let ((itemvar (gensym)))
  405.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  406.       (let ((bindlist (mapcar #'list SM1 SM2)))
  407.         (if bindlist
  408.           (push `(,itemvar ,item) bindlist)
  409.           (setq itemvar item)
  410.         )
  411.         (let ((valform `(ADJOIN ,itemvar ,SM5 ,@keylist)))
  412.           (if (simple-assignment-p SM4 SM3)
  413.             (setq SM4 (subst valform (first SM3) SM4))
  414.             (setq bindlist (nconc bindlist `((,(first SM3) ,valform))))
  415.           )
  416.           (if bindlist
  417.             `(LET* ,bindlist
  418.                ,SM4
  419.              )
  420.             SM4
  421.           )
  422. ) ) ) ) )
  423. ;-------------------------------------------------------------------------------
  424. (defmacro remf (place indicator &environment env)
  425.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  426.     (do* ((SM1r SM1 (cdr SM1r))
  427.           (SM2r SM2 (cdr SM2r))
  428.           (bindlist nil)
  429.           (indicatorvar (gensym))
  430.           (var1 (gensym))
  431.           (var2 (gensym)))
  432.          ((null SM1r)
  433.           (push `(,(first SM3) ,SM5) bindlist)
  434.           (push `(,indicatorvar ,indicator) bindlist)
  435.           `(LET* ,(nreverse bindlist)
  436.              (DO ((,var1 ,(first SM3) (CDDR ,var1))
  437.                   (,var2 NIL ,var1))
  438.                  ((ATOM ,var1) NIL)
  439.                (COND ((ATOM (CDR ,var1))
  440.                       (ERROR #+DEUTSCH "REMF: Property-Liste ungerader Länge aufgetreten."
  441.                              #+ENGLISH "REMF: property list with an odd length"
  442.                              #+FRANCAIS "REMF : Occurence d'une liste de propriétés de longueur impaire."
  443.                      ))
  444.                      ((EQ (CAR ,var1) ,indicatorvar)
  445.                       (IF ,var2
  446.                         (RPLACD (CDR ,var2) (CDDR ,var1))
  447.                         ,(let ((newvalform `(CDDR ,(first SM3))))
  448.                            (if (simple-assignment-p SM4 SM3)
  449.                              (subst newvalform (first SM3) SM4)
  450.                              `(PROGN (SETQ ,(first SM3) ,newvalform) ,SM4)
  451.                          ) )
  452.                       )
  453.                       (RETURN T)
  454.            ) ) )     )
  455.          )
  456.       (push `(,(first SM1r) ,(first SM2r)) bindlist)
  457. ) ) )
  458. ;-------------------------------------------------------------------------------
  459. (defmacro rotatef (&rest args &environment env)
  460.   (cond ((null args) NIL)
  461.         ((null (cdr args)) `(PROGN ,(car args) NIL) )
  462.         (t (do* ((arglist args (cdr arglist))
  463.                  (bindlist nil)
  464.                  (storelist nil)
  465.                  (lastvar nil)
  466.                  (firstbind nil))
  467.                 ((atom arglist)
  468.                  (setf (car firstbind) lastvar)
  469.                  `(LET* ,(nreverse bindlist) ,@(nreverse storelist) NIL)
  470.                 )
  471.              (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  472.                  (get-setf-method (first arglist) env)
  473.                (do* ((SM1r SM1 (cdr SM1r))
  474.                      (SM2r SM2 (cdr SM2r)))
  475.                     ((null SM1r))
  476.                  (push `(,(first SM1r) ,(first SM2r)) bindlist)
  477.                )
  478.                (push `(,lastvar ,SM5) bindlist)
  479.                (if (null firstbind) (setq firstbind (first bindlist)))
  480.                (push SM4 storelist)
  481.                (setq lastvar (first SM3))
  482. ) )     )  ) )
  483. ;-------------------------------------------------------------------------------
  484. (defmacro define-modify-macro (name lambdalist function &optional docstring)
  485.   (let* ((varlist nil)
  486.          (restvar nil))
  487.     (do* ((lambdalistr lambdalist (cdr lambdalistr))
  488.           (next))
  489.          ((null lambdalistr))
  490.       (setq next (first lambdalistr))
  491.       (cond ((eq next '&OPTIONAL))
  492.             ((eq next '&REST)
  493.              (if (symbolp (second lambdalistr))
  494.                (setq restvar (second lambdalistr))
  495.                (error #+DEUTSCH "In der Definition von ~S ist die &REST-Variable kein Symbol: ~S"
  496.                       #+ENGLISH "In the definition of ~S: &REST variable ~S should be a symbol."
  497.                       #+FRANCAIS "Dans la définition de ~S la variable pour &REST n'est pas un symbole : ~S."
  498.                       name (second lambdalistr)
  499.              ) )
  500.              (if (null (cddr lambdalistr))
  501.                (return)
  502.                (error #+DEUTSCH "Nach &REST ist nur eine Variable erlaubt; es kam: ~S"
  503.                       #+ENGLISH "Only one variable is allowed after &REST, not ~S"
  504.                       #+FRANCAIS "Une seule variable est permise pour &REST et non ~S."
  505.                       lambdalistr
  506.             )) )
  507.             ((or (eq next '&KEY) (eq next '&ALLOW-OTHER-KEYS) (eq next '&AUX))
  508.              (error #+DEUTSCH "In einer DEFINE-MODIFY-MACRO-Lambdaliste ist ~S unzulässig."
  509.                     #+ENGLISH "Illegal in a DEFINE-MODIFY-MACRO lambda list: ~S"
  510.                     #+FRANCAIS "~S n'est pas permis dans une liste lambda pour DEFINE-MODIFY-MACRO."
  511.                     next
  512.             ))
  513.             ((symbolp next) (push next varlist))
  514.             ((and (listp next) (symbolp (first next)))
  515.              (push (first next) varlist)
  516.             )
  517.             (t (error #+DEUTSCH "Lambdalisten dürfen nur Symbole und Listen enthalten, nicht aber ~S"
  518.                       #+ENGLISH "lambda list may only contain symbols and lists, not ~S"
  519.                       #+FRANCAIS "Les listes lambda ne peuvent contenir que des symboles et des listes et non ~S."
  520.                       next
  521.             )  )
  522.     ) )
  523.     (setq varlist (nreverse varlist))
  524.     `(DEFMACRO ,name (%REFERENCE ,@lambdalist &ENVIRONMENT ENV) ,docstring
  525.        (MULTIPLE-VALUE-BIND (DUMMIES VALS NEWVAL SETTER GETTER)
  526.            (GET-SETF-METHOD %REFERENCE ENV)
  527.          (DO ((D DUMMIES (CDR D))
  528.               (V VALS (CDR V))
  529.               (LET-LIST NIL (CONS (LIST (CAR D) (CAR V)) LET-LIST)))
  530.              ((NULL D)
  531.               (WHEN (SYMBOLP GETTER)
  532.                 (RETURN
  533.                   (SUBST
  534.                     (LIST* (QUOTE ,function) GETTER ,@varlist ,restvar)
  535.                     (CAR NEWVAL)
  536.                     SETTER
  537.               ) ) )
  538.               (PUSH
  539.                 (LIST
  540.                   (CAR NEWVAL)
  541.                   (IF (AND (LISTP %REFERENCE) (EQ (CAR %REFERENCE) 'THE))
  542.                     (LIST 'THE (CADR %REFERENCE)
  543.                       (LIST* (QUOTE ,function) GETTER ,@varlist ,restvar)
  544.                     )
  545.                     (LIST* (QUOTE ,function) GETTER ,@varlist ,restvar)
  546.                 ) )
  547.                 LET-LIST
  548.               )
  549.               (LIST 'LET* (NREVERSE LET-LIST) SETTER)
  550.      ) ) ) )
  551. ) )
  552. ;-------------------------------------------------------------------------------
  553. (define-modify-macro decf (&optional (delta 1)) -)
  554. ;-------------------------------------------------------------------------------
  555. (define-modify-macro incf (&optional (delta 1)) +)
  556. ;-------------------------------------------------------------------------------
  557. (defmacro setf (&whole form &rest args &environment env)
  558.   (let ((argcount (length args)))
  559.     (cond ((eql argcount 2)
  560.            (let* ((place (first args))
  561.                   (value (second args)))
  562.              (loop
  563.                ; 1. Schritt: nach globalen SETF-Definitionen suchen:
  564.                (when (and (consp place) (symbolp (car place)))
  565.                  (when (global-in-fenv-p (car place) (svref env 1))
  566.                    ; Operator nicht lokal definiert
  567.                    (let ((plist-info (get (first place) 'SYSTEM::SETF-EXPANDER)))
  568.                      (when plist-info
  569.                        (return-from setf
  570.                          (cond ((symbolp plist-info) ; Symbol kommt von kurzem DEFSETF
  571.                                 `(,plist-info ,@(cdr place) ,value)
  572.                                )
  573.                                ((and (eq (first place) 'THE) (eql (length place) 3))
  574.                                 `(SETF ,(third place) (THE ,(second place) ,value))
  575.                                )
  576.                                ((and (eq (first place) 'VALUES-LIST) (eql (length place) 2))
  577.                                 `(VALUES-LIST
  578.                                    (SETF ,(second place)
  579.                                          (MULTIPLE-VALUE-LIST ,value)
  580.                                ) ) )
  581.                                (t
  582.                                 (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  583.                                     (get-setf-method-multiple-value place env)
  584.                                   (declare (ignore SM5))
  585.                                   (do* ((SM1r SM1 (cdr SM1r))
  586.                                         (SM2r SM2 (cdr SM2r))
  587.                                         (bindlist nil))
  588.                                        ((null SM1r)
  589.                                         (if (eql (length SM3) 1) ; eine Store-Variable
  590.                                           `(LET* ,(nreverse
  591.                                                     (cons `(,(first SM3) ,value)
  592.                                                           bindlist
  593.                                                   ) )
  594.                                              ,SM4
  595.                                            )
  596.                                           ; mehrere Store-Variable
  597.                                           (if
  598.                                             ; Hat SM4 die Gestalt
  599.                                             ; (VALUES (SETQ v1 store1) ...) ?
  600.                                             (and (consp SM4) (eq (car SM4) 'VALUES)
  601.                                               (do ((SM3r SM3 (cdr SM3r))
  602.                                                    (SM4r (cdr SM4) (cdr SM4r)))
  603.                                                   ((or (null SM3r) (null SM4r))
  604.                                                    (and (null SM3r) (null SM4r))
  605.                                                   )
  606.                                                 (unless (simple-assignment-p (car SM4r) (list (car SM3r)))
  607.                                                   (return nil)
  608.                                             ) ) )
  609.                                             (let ((vlist (mapcar #'second (rest SM4))))
  610.                                               `(LET* ,(nreverse bindlist)
  611.                                                  (MULTIPLE-VALUE-SETQ ,vlist ,value)
  612.                                                  (VALUES ,@vlist)
  613.                                             )  )
  614.                                             `(LET* ,(nreverse bindlist)
  615.                                                (MULTIPLE-VALUE-BIND ,SM3 ,value
  616.                                                  ,SM4
  617.                                              ) )
  618.                                        )) )
  619.                                     (push `(,(first SM1r) ,(first SM2r)) bindlist)
  620.                        ) )     )) )
  621.                ) ) ) )
  622.                ; 2. Schritt: macroexpandieren
  623.                (when (eq place (setq place (macroexpand-1 place env)))
  624.                  (return)
  625.              ) )
  626.              ; 3. Schritt: Default-SETF-Methoden
  627.              (cond ((symbolp place)
  628.                     `(SETQ ,place ,value)
  629.                    )
  630.                    ((and (consp form) (symbolp (car form)))
  631.                     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  632.                         (get-setf-method-multiple-value place env)
  633.                       (declare (ignore SM5))
  634.                       ; SM4 hat die Gestalt `((SETF ,(first place)) ,@SM3 ,@SM1).
  635.                       ; SM3 ist überflüssig.
  636.                       `(LET* ,(mapcar #'list SM1 SM2)
  637.                          ,(subst value (first SM3) SM4)
  638.                        )
  639.                    ))
  640.                    (t (error #+DEUTSCH "Das ist keine erlaubte 'SETF-Place' : ~S"
  641.                              #+ENGLISH "Illegal SETF place: ~S"
  642.                              #+FRANCAIS "Ceci n'est pas une place modifiable valide : ~S"
  643.                              (first args)
  644.              )     )  )
  645.           ))
  646.           ((oddp argcount)
  647.            (error #+DEUTSCH "~S mit einer ungeraden Zahl von Argumenten aufgerufen: ~S"
  648.                   #+ENGLISH "~S called with an odd number of arguments: ~S"
  649.                   #+FRANCAIS "~S fut appelé avec un nombre impair d'arguments : ~S"
  650.                   'setf form
  651.           ))
  652.           (t (do* ((arglist args (cddr arglist))
  653.                    (L nil))
  654.                   ((null arglist) `(LET () (PROGN ,@(nreverse L))))
  655.                (push `(SETF ,(first arglist) ,(second arglist)) L)
  656.           )  )
  657. ) ) )
  658. ;-------------------------------------------------------------------------------
  659. (defmacro shiftf (&whole form &rest args &environment env)
  660.   (when (< (length args) 2)
  661.     (error #+DEUTSCH "SHIFTF mit zu wenig Argumenten aufgerufen: ~S"
  662.            #+ENGLISH "SHIFTF called with too few arguments: ~S"
  663.            #+FRANCAIS "SHIFTF fut appelé avec trop peu d'arguments : ~S"
  664.            form
  665.   ) )
  666.   (do* ((resultvar (gensym))
  667.         (arglist args (cdr arglist))
  668.         (bindlist nil)
  669.         (storelist nil)
  670.         (lastvar resultvar))
  671.        ((atom (cdr arglist))
  672.         (push `(,lastvar ,(first arglist)) bindlist)
  673.         `(LET* ,(nreverse bindlist) ,@(nreverse storelist) ,resultvar)
  674.        )
  675.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method (first arglist) env)
  676.       (do* ((SM1r SM1 (cdr SM1r))
  677.             (SM2r SM2 (cdr SM2r)))
  678.            ((null Sm1r))
  679.         (push `(,(first SM1r) ,(first SM2r)) bindlist)
  680.       )
  681.       (push `(,lastvar ,SM5) bindlist)
  682.       (push SM4 storelist)
  683.       (setq lastvar (first SM3))
  684. ) ) )
  685. ;-------------------------------------------------------------------------------
  686. ; Definition von places:
  687. ;-------------------------------------------------------------------------------
  688. (defsetf aref (array &rest indices) (value)
  689.   `(SYSTEM::STORE ,array ,@indices ,value)
  690. )
  691. ;-------------------------------------------------------------------------------
  692. (defun SYSTEM::%SETNTH (index list value)
  693.   (let ((pointer (nthcdr index list)))
  694.     (if (null pointer)
  695.       (error #+DEUTSCH "(SETF (NTH ...) ...) : Index ~S ist zu groß für ~S."
  696.              #+ENGLISH "(SETF (NTH ...) ...) : index ~S is too large for ~S"
  697.              #+FRANCAIS "(SETF (NTH ...) ...) : L'index ~S est trop grand pour ~S."
  698.              index list
  699.       )
  700.       (rplaca pointer value)
  701.     )
  702.     value
  703. ) )
  704. (defsetf nth SYSTEM::%SETNTH)
  705. ;-------------------------------------------------------------------------------
  706. (defsetf elt SYSTEM::%SETELT)
  707. ;-------------------------------------------------------------------------------
  708. (defsetf rest SYSTEM::%RPLACD)
  709. (defsetf first SYSTEM::%RPLACA)
  710. (defsetf second (list) (value) `(SYSTEM::%RPLACA (CDR ,list) ,value))
  711. (defsetf third (list) (value) `(SYSTEM::%RPLACA (CDDR ,list) ,value))
  712. (defsetf fourth (list) (value) `(SYSTEM::%RPLACA (CDDDR ,list) ,value))
  713. (defsetf fifth (list) (value) `(SYSTEM::%RPLACA (CDDDDR ,list) ,value))
  714. (defsetf sixth (list) (value) `(SYSTEM::%RPLACA (CDR (CDDDDR ,list)) ,value))
  715. (defsetf seventh (list) (value) `(SYSTEM::%RPLACA (CDDR (CDDDDR ,list)) ,value))
  716. (defsetf eighth (list) (value) `(SYSTEM::%RPLACA (CDDDR (CDDDDR ,list)) ,value))
  717. (defsetf ninth (list) (value) `(SYSTEM::%RPLACA (CDDDDR (CDDDDR ,list)) ,value))
  718. (defsetf tenth (list) (value) `(SYSTEM::%RPLACA (CDR (CDDDDR (CDDDDR ,list))) ,value))
  719.  
  720. (defsetf car SYSTEM::%RPLACA)
  721. (defsetf cdr SYSTEM::%RPLACD)
  722. (defsetf caar (list) (value) `(SYSTEM::%RPLACA (CAR ,list) ,value))
  723. (defsetf cadr (list) (value) `(SYSTEM::%RPLACA (CDR ,list) ,value))
  724. (defsetf cdar (list) (value) `(SYSTEM::%RPLACD (CAR ,list) ,value))
  725. (defsetf cddr (list) (value) `(SYSTEM::%RPLACD (CDR ,list) ,value))
  726. (defsetf caaar (list) (value) `(SYSTEM::%RPLACA (CAAR ,list) ,value))
  727. (defsetf caadr (list) (value) `(SYSTEM::%RPLACA (CADR ,list) ,value))
  728. (defsetf cadar (list) (value) `(SYSTEM::%RPLACA (CDAR ,list) ,value))
  729. (defsetf caddr (list) (value) `(SYSTEM::%RPLACA (CDDR ,list) ,value))
  730. (defsetf cdaar (list) (value) `(SYSTEM::%RPLACD (CAAR ,list) ,value))
  731. (defsetf cdadr (list) (value) `(SYSTEM::%RPLACD (CADR ,list) ,value))
  732. (defsetf cddar (list) (value) `(SYSTEM::%RPLACD (CDAR ,list) ,value))
  733. (defsetf cdddr (list) (value) `(SYSTEM::%RPLACD (CDDR ,list) ,value))
  734. (defsetf caaaar (list) (value) `(SYSTEM::%RPLACA (CAAAR ,list) ,value))
  735. (defsetf caaadr (list) (value) `(SYSTEM::%RPLACA (CAADR ,list) ,value))
  736. (defsetf caadar (list) (value) `(SYSTEM::%RPLACA (CADAR ,list) ,value))
  737. (defsetf caaddr (list) (value) `(SYSTEM::%RPLACA (CADDR ,list) ,value))
  738. (defsetf cadaar (list) (value) `(SYSTEM::%RPLACA (CDAAR ,list) ,value))
  739. (defsetf cadadr (list) (value) `(SYSTEM::%RPLACA (CDADR ,list) ,value))
  740. (defsetf caddar (list) (value) `(SYSTEM::%RPLACA (CDDAR ,list) ,value))
  741. (defsetf cadddr (list) (value) `(SYSTEM::%RPLACA (CDDDR ,list) ,value))
  742. (defsetf cdaaar (list) (value) `(SYSTEM::%RPLACD (CAAAR ,list) ,value))
  743. (defsetf cdaadr (list) (value) `(SYSTEM::%RPLACD (CAADR ,list) ,value))
  744. (defsetf cdadar (list) (value) `(SYSTEM::%RPLACD (CADAR ,list) ,value))
  745. (defsetf cdaddr (list) (value) `(SYSTEM::%RPLACD (CADDR ,list) ,value))
  746. (defsetf cddaar (list) (value) `(SYSTEM::%RPLACD (CDAAR ,list) ,value))
  747. (defsetf cddadr (list) (value) `(SYSTEM::%RPLACD (CDADR ,list) ,value))
  748. (defsetf cdddar (list) (value) `(SYSTEM::%RPLACD (CDDAR ,list) ,value))
  749. (defsetf cddddr (list) (value) `(SYSTEM::%RPLACD (CDDDR ,list) ,value))
  750. ;-------------------------------------------------------------------------------
  751. (defsetf svref SYSTEM::SVSTORE)
  752. ;-------------------------------------------------------------------------------
  753. (defsetf GET (symbol indicator &optional default) (value)
  754.   (let ((storeform `(SYSTEM::%PUT ,symbol ,indicator ,value)))
  755.     (if default
  756.       `(PROGN ,default ,storeform) ; default wird nur zum Schein ausgewertet
  757.       `,storeform
  758. ) ) )
  759. ;-------------------------------------------------------------------------------
  760. ; Schreibt zu einem bestimmten Indicator einen Wert in eine gegebene
  761. ; Propertyliste. Wert ist NIL falls erfolgreich getan oder die neue
  762. ; (erweiterte) Propertyliste.
  763. (defun sys::%putf (plist indicator value)
  764.   (do ((plistr plist (cddr plistr)))
  765.       ((atom plistr) (list* indicator value plist))
  766.     (when (atom (cdr plistr))
  767.       (error #+DEUTSCH "(SETF (GETF ...) ...) : Property-Liste ungerader Länge aufgetaucht."
  768.              #+ENGLISH "(SETF (GETF ...) ...) : property list with an odd length"
  769.              #+FRANCAIS "(SETF (GETF ...) ...) : Occurence d'une liste de propriétés de longueur impaire."
  770.     ))
  771.     (when (eq (car plistr) indicator)
  772.       (rplaca (cdr plistr) value)
  773.       (return nil)
  774. ) ) )
  775. (define-setf-method getf (place indicator &optional default &environment env)
  776.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  777.     (let* ((storevar (gensym))
  778.            (indicatorvar (gensym))
  779.            (defaultvar-list (if default (list (gensym)) `()))
  780.           )
  781.       (values
  782.         `(,@SM1 ,indicatorvar ,@defaultvar-list)
  783.         `(,@SM2 ,indicator    ,@(if default `(,default) `()))
  784.         `(,storevar)
  785.         `(LET ((,(first SM3) (SYS::%PUTF ,SM5 ,indicatorvar ,storevar)))
  786.            ,@defaultvar-list ; defaultvar zum Schein auswerten
  787.            (WHEN ,(first SM3) ,SM4)
  788.            ,storevar
  789.          )
  790.         `(GETF ,SM5 ,indicatorvar ,@defaultvar-list)
  791. ) ) ) )
  792. ;-------------------------------------------------------------------------------
  793. (defsetf GETHASH (key hashtable &optional default) (value)
  794.   (let ((storeform `(SYSTEM::PUTHASH ,key ,hashtable ,value)))
  795.     (if default
  796.       `(PROGN ,default ,storeform) ; default wird nur zum Schein ausgewertet
  797.       `,storeform
  798. ) ) )
  799. ;-------------------------------------------------------------------------------
  800. #| ; siehe oben:
  801. (defun SYSTEM::%SET-DOCUMENTATION (symbol doctype value)
  802.   (unless (function-name-p symbol)
  803.     (error #+DEUTSCH "Das ist als erstes Argument unzulässig, da kein Symbol: ~S"
  804.            #+ENGLISH "first argument ~S is illegal, not a symbol"
  805.            #+FRANCAIS "Le premier argument ~S est invalide car ce n'est pas un symbole."
  806.            symbol
  807.   ) )
  808.   (setq symbol (get-funname-symbol symbol))
  809.   (if (null value)
  810.     (progn (remf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype) nil)
  811.     (setf (getf (get symbol 'SYSTEM::DOCUMENTATION-STRINGS) doctype) value)
  812. ) )
  813. |#
  814. (defsetf documentation SYSTEM::%SET-DOCUMENTATION)
  815. ;-------------------------------------------------------------------------------
  816. (defsetf fill-pointer SYSTEM::SET-FILL-POINTER)
  817. ;-------------------------------------------------------------------------------
  818. (defsetf readtable-case SYSTEM::SET-READTABLE-CASE)
  819. ;-------------------------------------------------------------------------------
  820. (defsetf SYMBOL-VALUE SET)
  821. ;-------------------------------------------------------------------------------
  822. (defsetf SYMBOL-FUNCTION SYSTEM::%PUTD)
  823. ;-------------------------------------------------------------------------------
  824. (defsetf SYMBOL-PLIST SYSTEM::%PUTPLIST)
  825. ;-------------------------------------------------------------------------------
  826. (defun SYSTEM::SET-FDEFINITION (name value)
  827.   (setf (symbol-function (get-funname-symbol name)) value)
  828. )
  829. (defsetf FDEFINITION SYSTEM::SET-FDEFINITION)
  830. ;-------------------------------------------------------------------------------
  831. (defsetf MACRO-FUNCTION (symbol) (value)
  832.   `(PROGN
  833.      (SETF (SYMBOL-FUNCTION ,symbol) (CONS 'SYSTEM::MACRO ,value))
  834.      (REMPROP ,symbol 'SYSTEM::MACRO)
  835.      ,value
  836.    )
  837. )
  838. ;-------------------------------------------------------------------------------
  839. (defsetf CHAR SYSTEM::STORE-CHAR)
  840. (defsetf SCHAR SYSTEM::STORE-SCHAR)
  841. (defsetf BIT SYSTEM::STORE)
  842. (defsetf SBIT SYSTEM::STORE)
  843. (defsetf SUBSEQ (sequence start &optional end) (value)
  844.   `(PROGN (REPLACE ,sequence ,value :START1 ,start :END1 ,end) ,value)
  845. )
  846. ;-------------------------------------------------------------------------------
  847. (define-setf-method char-bit (char name &environment env)
  848.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method char env)
  849.     (let* ((namevar (gensym))
  850.            (storevar (gensym)))
  851.       (values `(,@SM1 ,namevar)
  852.               `(,@SM2 ,name)
  853.               `(,storevar)
  854.               `(LET ((,(first SM3) (SET-CHAR-BIT ,SM5 ,namevar ,storevar)))
  855.                  ,SM4
  856.                  ,storevar
  857.                )
  858.               `(CHAR-BIT ,SM5 ,namevar)
  859. ) ) ) )
  860. ;-------------------------------------------------------------------------------
  861. (define-setf-method LDB (bytespec integer &environment env)
  862.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method integer env)
  863.     (let* ((bytespecvar (gensym))
  864.            (storevar (gensym)))
  865.       (values (cons bytespecvar SM1)
  866.               (cons bytespec SM2)
  867.               `(,storevar)
  868.               `(LET ((,(first SM3) (DPB ,storevar ,bytespecvar ,SM5)))
  869.                  ,SM4
  870.                  ,storevar
  871.                )
  872.               `(LDB ,bytespecvar ,SM5)
  873. ) ) ) )
  874. ;-------------------------------------------------------------------------------
  875. (define-setf-method MASK-FIELD (bytespec integer &environment env)
  876.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method integer env)
  877.     (let* ((bytespecvar (gensym))
  878.            (storevar (gensym)))
  879.       (values (cons bytespecvar SM1)
  880.               (cons bytespec SM2)
  881.               `(,storevar)
  882.               `(LET ((,(first SM3) (DEPOSIT-FIELD ,storevar ,bytespecvar ,SM5)))
  883.                  ,SM4
  884.                  ,storevar
  885.                )
  886.               `(MASK-FIELD ,bytespecvar ,SM5)
  887. ) ) ) )
  888. ;-------------------------------------------------------------------------------
  889. (define-setf-method THE (type place &environment env)
  890.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method place env)
  891.     (values SM1 SM2 SM3
  892.             (subst `(THE ,type ,(first SM3)) (first SM3) SM4)
  893.             `(THE ,type ,SM5)
  894. ) ) )
  895. ;-------------------------------------------------------------------------------
  896. (define-setf-method APPLY (fun &rest args &environment env)
  897.   (if (and (listp fun)
  898.            (eq (list-length fun) 2)
  899.            (eq (first fun) 'FUNCTION)
  900.            (symbolp (second fun))
  901.       )
  902.     (setq fun (second fun))
  903.     (error #+DEUTSCH "SETF von APPLY ist nur für Funktionen der Form #'symbol als Argument definiert."
  904.            #+ENGLISH "SETF APPLY is only defined for functions of the form #'symbol."
  905.            #+FRANCAIS "Un SETF de APPLY n'est défini que pour les fonctions de la forme #'symbole."
  906.   ) )
  907.   (multiple-value-bind (SM1 SM2 SM3 SM4 SM5) (get-setf-method (cons fun args) env)
  908.     (unless (eq (car (last args)) (car (last SM2)))
  909.       (error #+DEUTSCH "APPLY von ~S kann nicht als 'SETF-Place' aufgefaßt werden."
  910.              #+ENGLISH "APPLY on ~S is not a SETF place."
  911.              #+FRANCAIS "APPLY de ~S ne peux pas être considéré comme une place modifiable."
  912.              fun
  913.     ) )
  914.     (let ((item (car (last SM1)))) ; 'item' steht für eine Argumentliste!
  915.       (labels ((splice (arglist)
  916.                  ; Würde man in (LIST . arglist) das 'item' nicht als 1 Element,
  917.                  ; sondern gespliced, sozusagen als ',@item', haben wollen, so
  918.                  ; bräuchte man die Form, die (splice arglist) liefert.
  919.                  (if (endp arglist)
  920.                    'NIL
  921.                    (let ((rest (splice (cdr arglist))))
  922.                      (if (eql (car arglist) item)
  923.                        ; ein (APPEND item ...) davorhängen, wie bei Backquote
  924.                        (backquote-append item rest)
  925.                        ; ein (CONS (car arglist) ...) davorhängen, wie bei Backquote
  926.                        (backquote-cons (car arglist) rest)
  927.               )) ) ) )
  928.         (flet ((call-splicing (form)
  929.                  ; ersetzt einen Funktionsaufruf form durch einen, bei dem
  930.                  ; 'item' nicht 1 Argument, sondern eine Argumentliste liefert
  931.                  (let ((fun (first form))
  932.                        (argform (splice (rest form))))
  933.                    ; (APPLY #'fun argform) vereinfachen:
  934.                    ; (APPLY #'fun NIL) --> (fun)
  935.                    ; (APPLY #'fun (LIST ...)) --> (fun ...)
  936.                    ; (APPLY #'fun (CONS x y)) --> (APPLY #'fun x y)
  937.                    ; (APPLY #'fun (LIST* ... z)) --> (APPLY #'fun ... z)
  938.                    (if (or (null argform)
  939.                            (and (consp argform) (eq (car argform) 'LIST))
  940.                        )
  941.                      (cons fun (cdr argform))
  942.                      (list* 'APPLY
  943.                             (list 'FUNCTION fun)
  944.                             (if (and (consp argform)
  945.                                      (or (eq (car argform) 'LIST*)
  946.                                          (eq (car argform) 'CONS)
  947.                                 )    )
  948.                               (cdr argform)
  949.                               (list argform)
  950.               )) ) ) )      )
  951.           (values SM1 SM2 SM3 (call-splicing SM4) (call-splicing SM5))
  952. ) ) ) ) )
  953. ;-------------------------------------------------------------------------------
  954. ; Zusätzliche Definitionen von places
  955. ;-------------------------------------------------------------------------------
  956. (define-setf-method funcall (fun &rest args &environment env)
  957.   (unless (and (listp fun)
  958.                (eq (list-length fun) 2)
  959.                (let ((fun1 (first fun)))
  960.                  (or (eq fun1 'FUNCTION) (eq fun1 'QUOTE))
  961.                )
  962.                (symbolp (second fun))
  963.                (setq fun (second fun))
  964.           )
  965.     (error #+DEUTSCH "SETF von FUNCALL ist nur für Funktionen der Form #'symbol definiert."
  966.            #+ENGLISH "SETF FUNCALL is only defined for functions of the form #'symbol."
  967.            #+FRANCAIS "Un SETF de FUNCALL n'est défini que pour les fonctions de la forme #'symbole."
  968.   ) )
  969.   (get-setf-method (cons fun args) env)
  970. )
  971. ;-------------------------------------------------------------------------------
  972. (defsetf GET-DISPATCH-MACRO-CHARACTER
  973.          (disp-char sub-char &optional (readtable '*READTABLE*)) (value)
  974.   `(PROGN (SET-DISPATCH-MACRO-CHARACTER ,disp-char ,sub-char ,value ,readtable) ,value)
  975. )
  976. ;-------------------------------------------------------------------------------
  977. (defsetf long-float-digits SYSTEM::%SET-LONG-FLOAT-DIGITS)
  978. ;-------------------------------------------------------------------------------
  979. (defsetf DEFAULT-DIRECTORY () (value)
  980.   `(PROGN (CD ,value) ,value)
  981. )
  982. ;-------------------------------------------------------------------------------
  983. ; Handhabung von (SETF (VALUES place1 ... placek) form)
  984. ; --> (MULTIPLE-VALUE-BIND (dummy1 ... dummyk) form
  985. ;       (SETF place1 dummy1 ... placek dummyk)
  986. ;       (VALUES dummy1 ... dummyk)
  987. ;     )
  988. (define-setf-method VALUES (&rest subplaces &environment env)
  989.   (multiple-value-bind (temps vals stores storeforms accessforms)
  990.       (setf-VALUES-aux subplaces env)
  991.     (values temps
  992.             vals
  993.             stores
  994.             `(VALUES ,@storeforms)
  995.             `(VALUES ,@accessforms)
  996. ) ) )
  997. (defun setf-VALUES-aux (places env)
  998.   (do ((temps nil)
  999.        (vals nil)
  1000.        (stores nil)
  1001.        (storeforms nil)
  1002.        (accessforms nil)
  1003.        (placesr places))
  1004.       ((atom placesr)
  1005.        (setq temps (nreverse temps))
  1006.        (setq vals (nreverse vals))
  1007.        (setq stores (nreverse stores))
  1008.        (setq storeforms (nreverse storeforms))
  1009.        (setq accessforms (nreverse accessforms))
  1010.        (values temps vals stores storeforms accessforms)
  1011.       )
  1012.     (multiple-value-bind (SM1 SM2 SM3 SM4 SM5)
  1013.         (get-setf-method (pop placesr) env)
  1014.       (setq temps (revappend SM1 temps))
  1015.       (setq vals (revappend SM2 vals))
  1016.       (setq stores (revappend SM3 stores))
  1017.       (setq storeforms (cons SM4 storeforms))
  1018.       (setq accessforms (cons SM5 accessforms))
  1019. ) ) )
  1020. ;-------------------------------------------------------------------------------
  1021. ; Analog zu (MULTIPLE-VALUE-SETQ (var1 ... vark) form) :
  1022. ; (MULTIPLE-VALUE-SETF (place1 ... placek) form)
  1023. ; --> (VALUES (SETF (VALUES place1 ... placek) form))
  1024. ; --> (MULTIPLE-VALUE-BIND (dummy1 ... dummyk) form
  1025. ;       (SETF place1 dummy1 ... placek dummyk)
  1026. ;       dummy1
  1027. ;     )
  1028. (defmacro multiple-value-setf (places form &environment env)
  1029.   (multiple-value-bind (temps vals stores storeforms accessforms)
  1030.       (setf-VALUES-aux places env)
  1031.     (declare (ignore accessforms))
  1032.     `(LET* ,(mapcar #'list temps vals)
  1033.        (MULTIPLE-VALUE-BIND ,stores ,form
  1034.          ,@storeforms
  1035.          (first stores) ; (null stores) -> NIL -> Wert NIL
  1036.      ) )
  1037. ) )
  1038. ;-------------------------------------------------------------------------------
  1039.  
  1040.